SingleTablePolymorphismPersister.java
package org.codefilarete.stalactite.engine.runtime.singletable;
import javax.annotation.Nullable;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Queue;
import java.util.Set;
import java.util.function.BiFunction;
import java.util.function.Function;
import java.util.stream.Collectors;
import org.codefilarete.reflection.Accessor;
import org.codefilarete.stalactite.engine.DeleteExecutor;
import org.codefilarete.stalactite.engine.EntityPersister;
import org.codefilarete.stalactite.engine.InsertExecutor;
import org.codefilarete.stalactite.dsl.PolymorphismPolicy.SingleTablePolymorphism;
import org.codefilarete.stalactite.engine.SelectExecutor;
import org.codefilarete.stalactite.engine.UpdateExecutor;
import org.codefilarete.stalactite.engine.configurer.onetomany.OneToManyRelationConfigurer;
import org.codefilarete.stalactite.engine.runtime.AbstractPolymorphismPersister;
import org.codefilarete.stalactite.engine.runtime.ConfiguredRelationalPersister;
import org.codefilarete.stalactite.engine.runtime.EntityMappingWrapper;
import org.codefilarete.stalactite.engine.runtime.FirstPhaseRelationLoader;
import org.codefilarete.stalactite.engine.runtime.PersisterWrapper;
import org.codefilarete.stalactite.engine.runtime.PolymorphicPersister;
import org.codefilarete.stalactite.engine.runtime.RelationIds;
import org.codefilarete.stalactite.engine.runtime.RelationalEntityPersister;
import org.codefilarete.stalactite.engine.runtime.SecondPhaseRelationLoader;
import org.codefilarete.stalactite.engine.runtime.load.EntityInflater.EntityMappingAdapter;
import org.codefilarete.stalactite.engine.runtime.load.EntityJoinTree;
import org.codefilarete.stalactite.engine.runtime.load.EntityJoinTree.JoinType;
import org.codefilarete.stalactite.engine.runtime.load.JoinNode;
import org.codefilarete.stalactite.engine.runtime.load.SingleTablePolymorphicRelationJoinNode;
import org.codefilarete.stalactite.mapping.EntityMapping;
import org.codefilarete.stalactite.mapping.IdMapping;
import org.codefilarete.stalactite.mapping.Mapping.ShadowColumnValueProvider;
import org.codefilarete.stalactite.mapping.RowTransformer.TransformerListener;
import org.codefilarete.stalactite.query.model.Selectable;
import org.codefilarete.stalactite.sql.ConnectionProvider;
import org.codefilarete.stalactite.sql.Dialect;
import org.codefilarete.stalactite.sql.ddl.structure.Column;
import org.codefilarete.stalactite.sql.ddl.structure.Key;
import org.codefilarete.stalactite.sql.ddl.structure.PrimaryKey;
import org.codefilarete.stalactite.sql.ddl.structure.Table;
import org.codefilarete.stalactite.sql.result.BeanRelationFixer;
import org.codefilarete.stalactite.sql.result.ColumnedRow;
import org.codefilarete.tool.Duo;
import org.codefilarete.tool.bean.Objects;
import org.codefilarete.tool.collection.Arrays;
import org.codefilarete.tool.collection.Collections;
import org.codefilarete.tool.collection.Iterables;
import org.codefilarete.tool.collection.KeepOrderMap;
import org.codefilarete.tool.collection.KeepOrderSet;
import static org.codefilarete.stalactite.engine.runtime.load.EntityJoinTree.ROOT_JOIN_NAME;
/**
* @author Guillaume Mary
*/
public class SingleTablePolymorphismPersister<C, I, T extends Table<T>, DTYPE> extends AbstractPolymorphismPersister<C, I> {
@SuppressWarnings("java:S5164" /* remove() is called by SecondPhaseRelationLoader.afterSelect() */)
private static final ThreadLocal<Queue<Set<RelationIds<Object /* E */, Object /* target */, Object /* target identifier */ >>>> DIFFERED_ENTITY_LOADER = new ThreadLocal<>();
private final Column<T, DTYPE> discriminatorColumn;
private final SingleTablePolymorphism<C, DTYPE> polymorphismPolicy;
public SingleTablePolymorphismPersister(ConfiguredRelationalPersister<C, I> mainPersister,
Map<? extends Class<C>, ? extends ConfiguredRelationalPersister<C, I>> subEntitiesPersisters,
ConnectionProvider connectionProvider,
Dialect dialect,
Column<T, DTYPE> discriminatorColumn,
SingleTablePolymorphism<C, DTYPE> polymorphismPolicy) {
super(mainPersister,
subEntitiesPersisters,
new SingleTablePolymorphismEntityFinder<>(mainPersister,
subEntitiesPersisters,
discriminatorColumn,
polymorphismPolicy,
connectionProvider,
dialect));
this.discriminatorColumn = discriminatorColumn;
this.polymorphismPolicy = polymorphismPolicy;
ShadowColumnValueProvider<C, T> discriminatorValueProvider = new ShadowColumnValueProvider<C, T>() {
@Override
public Set<Column<T, ?>> getColumns() {
return Arrays.asHashSet(discriminatorColumn);
}
@Override
public Map<Column<T, ?>, ?> giveValue(C bean) {
Map<Column<T, ?>, Object> result = new HashMap<>();
result.put(discriminatorColumn, polymorphismPolicy.getDiscriminatorValue((Class<? extends C>) bean.getClass()));
return result;
}
};
this.subEntitiesPersisters.values().forEach(subclassPersister -> ((EntityMapping) subclassPersister.getMapping())
.addShadowColumnInsert(discriminatorValueProvider));
subEntitiesPersisters.forEach((type, persister) ->
mainPersister.copyRootJoinsTo(persister.getEntityJoinTree(), ROOT_JOIN_NAME)
);
}
@Override
public EntityJoinTree<C, I> getEntityJoinTree() {
return ((SingleTablePolymorphismEntityFinder) entityFinder).getEntityJoinTree();
}
@Override
public <LEFTTABLE extends Table<LEFTTABLE>, SUBTABLE extends Table<SUBTABLE>, JOINTYPE> void propagateMappedAssociationToSubTables(
Key<SUBTABLE, JOINTYPE> foreignKey,
PrimaryKey<LEFTTABLE, JOINTYPE> leftPrimaryKey,
BiFunction<Key<SUBTABLE, JOINTYPE>, PrimaryKey<LEFTTABLE, JOINTYPE>, String> foreignKeyNamingFunction) {
SUBTABLE subTable = mainPersister.getMainTable();
Key.KeyBuilder<SUBTABLE, JOINTYPE> projectedKeyBuilder = Key.from(subTable);
((Set<Column<SUBTABLE, ?>>) foreignKey.getColumns()).forEach(column -> {
projectedKeyBuilder.addColumn(subTable.addColumn(column.getName(), column.getJavaType(), column.getSize(), column.isNullable()));
});
Key<SUBTABLE, JOINTYPE> projectedKey = projectedKeyBuilder.build();
mainPersister.getEntityJoinTree().addPassiveJoin(EntityJoinTree.ROOT_JOIN_NAME, foreignKey, projectedKey, EntityJoinTree.JoinType.INNER, java.util.Collections.emptySet());
subTable.addForeignKey(foreignKeyNamingFunction, projectedKey, leftPrimaryKey);
}
@Override
public Set<Class<? extends C>> getSupportedEntityTypes() {
Set<Class<? extends C>> result = new HashSet<>();
this.subEntitiesPersisters.forEach((c, p) -> {
if (p instanceof PolymorphicPersister) {
result.addAll((Collection) ((PolymorphicPersister<?>) p).getSupportedEntityTypes());
} else if (p instanceof PersisterWrapper && ((PersisterWrapper<C, I>) p).getDeepestDelegate() instanceof PolymorphicPersister) {
result.addAll(((PolymorphicPersister) ((PersisterWrapper) p).getDeepestDelegate()).getSupportedEntityTypes());
} else {
result.add(c);
}
});
return result;
}
@Override
public Collection<Table<?>> giveImpliedTables() {
// Implied tables are those of sub entities.
// Note that doing this lately (not in constructor) guaranties that it is uptodate because sub entities may have relations which are configured
// out of constructor by caller
Set<Table<?>> subTables = subEntitiesPersisters.values().stream().flatMap(p -> p.giveImpliedTables().stream()).collect(Collectors.toSet());
return Collections.cat(mainPersister.giveImpliedTables(), subTables);
}
@Override
public void doInsert(Iterable<? extends C> entities) {
Map<EntityPersister<C, I>, Set<C>> entitiesPerType = computeEntitiesPerPersister(entities);
entitiesPerType.forEach(InsertExecutor::insert);
}
@Override
public void doUpdateById(Iterable<? extends C> entities) {
Map<EntityPersister<C, I>, Set<C>> entitiesPerType = computeEntitiesPerPersister(entities);
entitiesPerType.forEach(UpdateExecutor::updateById);
}
@Override
public void doUpdate(Iterable<? extends Duo<C, C>> differencesIterable, boolean allColumnsStatement) {
// Below we keep the order of given entities mainly to get steady unit tests. Meanwhile, this may have performance
// impacts but it's very difficult to measure
Map<UpdateExecutor<C>, Set<Duo<C, C>>> entitiesPerType = new KeepOrderMap<>();
differencesIterable.forEach(payload ->
this.subEntitiesPersisters.values().forEach(persister -> {
C entity = Objects.preventNull(payload.getLeft(), payload.getRight());
if (persister.getClassToPersist().isInstance(entity)) {
entitiesPerType.computeIfAbsent(persister, p -> new KeepOrderSet<>()).add(payload);
}
})
);
entitiesPerType.forEach((updateExecutor, adhocEntities) -> updateExecutor.update(adhocEntities, allColumnsStatement));
}
@Override
public void doDelete(Iterable<? extends C> entities) {
Map<EntityPersister<C, I>, Set<C>> entitiesPerType = computeEntitiesPerPersister(entities);
entitiesPerType.forEach(DeleteExecutor::delete);
}
@Override
public void doDeleteById(Iterable<? extends C> entities) {
Map<EntityPersister<C, I>, Set<C>> entitiesPerType = computeEntitiesPerPersister(entities);
entitiesPerType.forEach(DeleteExecutor::deleteById);
}
private Map<EntityPersister<C, I>, Set<C>> computeEntitiesPerPersister(Iterable<? extends C> entities) {
Map<EntityPersister<C, I>, Set<C>> entitiesPerType = new KeepOrderMap<>();
entities.forEach(entity ->
this.subEntitiesPersisters.values().forEach(persister -> {
if (persister.getClassToPersist().isInstance(entity)) {
entitiesPerType.computeIfAbsent(persister, p -> new KeepOrderSet<>()).add(entity);
}
})
);
return entitiesPerType;
}
@Override
public <E, ID> void copyRootJoinsTo(EntityJoinTree<E, ID> entityJoinTree, String joinName) {
throw new UnsupportedOperationException();
}
/**
* Overridden to capture {@link EntityMapping#addShadowColumnInsert(ShadowColumnValueProvider)} and
* {@link EntityMapping#addShadowColumnUpdate(ShadowColumnValueProvider)} (see {@link OneToManyRelationConfigurer})
* Made to dispatch those methods subclass strategies since their persisters are in charge of managing their entities (not the parent one).
*
* Design question : one may think that's not a good design to override a getter, caller should invoke an intention-clear method on
* ourselves (Persister) but the case is to add a silent Column insert/update which is not the goal of the Persister to know implementation
* detail : they are to manage cascades and coordinate their mapping strategies. {@link EntityMapping} are in charge of knowing
* {@link Column} actions.
*
* @return an enhanced version of our main persister mapping strategy which dispatches silent column insert/update to sub-entities ones
*/
@Override
public EntityMapping<C, I, T> getMapping() {
return new EntityMappingWrapper<C, I, T>(mainPersister.getMapping()) {
@Override
public void addTransformerListener(TransformerListener<C> listener) {
subEntitiesPersisters.values().forEach(p -> ((EntityMapping) p.getMapping()).addTransformerListener(listener));
}
@Override
public void addShadowColumnInsert(ShadowColumnValueProvider<C, T> provider) {
subEntitiesPersisters.values().forEach(p -> ((EntityMapping) p.getMapping()).addShadowColumnInsert(provider));
}
@Override
public void addShadowColumnUpdate(ShadowColumnValueProvider<C, T> provider) {
subEntitiesPersisters.values().forEach(p -> ((EntityMapping) p.getMapping()).addShadowColumnUpdate(provider));
}
};
}
@Override
public <SRC, T1 extends Table<T1>, T2 extends Table<T2>, SRCID, JOINID> String joinAsOne(RelationalEntityPersister<SRC, SRCID> sourcePersister,
Accessor<SRC, C> propertyAccessor,
Key<T1, JOINID> leftColumn,
Key<T2, JOINID> rightColumn,
String rightTableAlias,
BeanRelationFixer<SRC, C> beanRelationFixer,
boolean optional,
boolean loadSeparately) {
if (loadSeparately) {
SingleTableFirstPhaseRelationLoader singleTableFirstPhaseRelationLoader = new SingleTableFirstPhaseRelationLoader(mainPersister.getMapping().getIdMapping(),
this,
(ThreadLocal<Queue<Set<RelationIds<Object, C, I>>>>) (ThreadLocal) DIFFERED_ENTITY_LOADER,
discriminatorColumn, subEntitiesPersisters::get);
String createdJoinNodeName = sourcePersister.getEntityJoinTree().addMergeJoin(ROOT_JOIN_NAME,
singleTableFirstPhaseRelationLoader,
leftColumn, rightColumn, optional ? JoinType.OUTER : JoinType.INNER);
// adding second phase loader
sourcePersister.addSelectListener(new SecondPhaseRelationLoader<>(beanRelationFixer, DIFFERED_ENTITY_LOADER));
return createdJoinNodeName;
} else {
return join(
sourcePersister.getEntityJoinTree(),
ROOT_JOIN_NAME,
mainPersister,
propertyAccessor,
leftColumn,
rightColumn,
new HashSet<>(this.subEntitiesPersisters.values()),
beanRelationFixer,
polymorphismPolicy,
(Column<T2, DTYPE>) discriminatorColumn);
}
}
@Override
public <SRC, T1 extends Table<T1>, T2 extends Table<T2>, SRCID, JOINID> String joinAsMany(String joinName,
RelationalEntityPersister<SRC, SRCID> sourcePersister,
Accessor<SRC, ?> propertyAccessor,
Key<T1, JOINID> leftColumn,
Key<T2, JOINID> rightColumn,
BeanRelationFixer<SRC, C> beanRelationFixer,
@Nullable Function<ColumnedRow, Object> duplicateIdentifierProvider,
Set<? extends Column<T2, ?>> selectableColumns,
boolean optional,
boolean loadSeparately) {
if (loadSeparately) {
// Subgraph loading is made in 2 phases (load ids, then entities in a second SQL request done by load listener)
SingleTableFirstPhaseRelationLoader singleTableFirstPhaseRelationLoader = new SingleTableFirstPhaseRelationLoader(mainPersister.getMapping().getIdMapping(),
this,
(ThreadLocal<Queue<Set<RelationIds<Object, C, I>>>>) (ThreadLocal) DIFFERED_ENTITY_LOADER,
discriminatorColumn, subEntitiesPersisters::get);
String createdJoinNodeName = sourcePersister.getEntityJoinTree().addMergeJoin(joinName,
singleTableFirstPhaseRelationLoader,
leftColumn, rightColumn, JoinType.OUTER);
// adding second phase loader
sourcePersister.addSelectListener(new SecondPhaseRelationLoader<>(beanRelationFixer, DIFFERED_ENTITY_LOADER));
return createdJoinNodeName;
} else {
return join(
sourcePersister.getEntityJoinTree(),
joinName,
mainPersister,
propertyAccessor,
leftColumn,
rightColumn,
new HashSet<>(this.subEntitiesPersisters.values()),
beanRelationFixer,
polymorphismPolicy,
(Column<T2, DTYPE>) discriminatorColumn);
}
}
private <SRC, SRCID, U extends C, T1 extends Table<T1>, T2 extends Table<T2>, ID, JOINCOLTYPE> String join(
EntityJoinTree<SRC, SRCID> entityJoinTree,
String leftStrategyName,
ConfiguredRelationalPersister<U, ID> mainPersister,
Accessor<SRC, ?> propertyAccessor,
Key<T1, JOINCOLTYPE> leftJoinColumn,
Key<T2, JOINCOLTYPE> rightJoinColumn,
Set<ConfiguredRelationalPersister<? extends U, ID>> subPersisters,
BeanRelationFixer<SRC, U> beanRelationFixer,
SingleTablePolymorphism<U, DTYPE> polymorphismPolicy,
Column<T2, DTYPE> discriminatorColumn) {
return entityJoinTree.addJoin(leftStrategyName, parent -> new SingleTablePolymorphicRelationJoinNode<>(
(JoinNode<SRC, T1>) (JoinNode) parent,
propertyAccessor,
leftJoinColumn,
rightJoinColumn,
JoinType.OUTER,
mainPersister.getMainTable().getColumns(),
null,
new EntityMappingAdapter<>(mainPersister.<T>getMapping()),
(BeanRelationFixer<Object, U>) beanRelationFixer,
discriminatorColumn,
subPersisters,
polymorphismPolicy));
}
private class SingleTableFirstPhaseRelationLoader extends FirstPhaseRelationLoader<C, I> {
private final Column<T, DTYPE> discriminatorColumn;
private final Function<Class, SelectExecutor> subtypeSelectors;
private final Set<DTYPE> discriminatorValues;
private SingleTableFirstPhaseRelationLoader(IdMapping<C, I> subEntityIdMapping,
SelectExecutor<C, I> selectExecutor,
ThreadLocal<Queue<Set<RelationIds<Object, C, I>>>> relationIdsHolder,
Column<T, DTYPE> discriminatorColumn, Function<Class, SelectExecutor> subtypeSelectors) {
// Note that selectExecutor won't be used because we dynamically lookup for it in fillCurrentRelationIds
super(subEntityIdMapping, selectExecutor, relationIdsHolder);
this.discriminatorColumn = discriminatorColumn;
this.subtypeSelectors = subtypeSelectors;
this.discriminatorValues = Iterables.collect(polymorphismPolicy.getSubClasses(), conf -> polymorphismPolicy.getDiscriminatorValue(conf.getEntityType()), HashSet::new);
}
@Override
protected void fillCurrentRelationIds(ColumnedRow columnedRow, Object bean) {
DTYPE discriminator = columnedRow.get(discriminatorColumn);
// we avoid NPE on polymorphismPolicy.getClass(discriminator) caused by null discriminator in case of empty relation
// by only treating known discriminator values (preferred way to check against null because type can be primitive one)
if (discriminatorValues.contains(discriminator)) {
Set<RelationIds<Object, C, I>> relationIds = relationIdsHolder.get().peek();
I id = idMapping.getIdentifierAssembler().assemble(columnedRow);
relationIds.add(new RelationIds<>(giveSelector(discriminator), idMapping.getIdAccessor()::getId, bean, id));
}
}
@Override
public Set<Selectable<?>> getSelectableColumns() {
Set<Selectable<?>> result = new HashSet<>(idMapping.getIdentifierAssembler().getColumns());
result.add((Selectable) discriminatorColumn);
return result;
}
private SelectExecutor giveSelector(DTYPE discriminator) {
return subtypeSelectors.apply(polymorphismPolicy.getClass(discriminator));
}
}
}